package com.lyn.gulimall.search.thread;

import java.util.concurrent.*;

public class ThreadTest {
    public static ExecutorService executor = Executors.newFixedThreadPool(10);


    public static void main(String[] args) throws ExecutionException, InterruptedException {
        System.out.println("main……start……");
//        CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 2;
//            System.out.println("运行结果：" + i);
//        }, executor);

        /**
         * 方法完成后的感知
         */
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 0;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).whenComplete((res,exception)->{
//            //虽然能得到异常信息，但是没法修改返回数据
//            System.out.println("异步能武成功完成了……结果是："+res+";异常时："+exception);
//        }).exceptionally(throwable -> {
//            //可以感知异常，同时返回默认值
//            return 10;
//        });

        /**
         * 方法执行完成后的处理
         */
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 2;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).handle((t,u)->{
//            if(t!=null){
//                return t*2;
//            }
//            if(u!=null){
//                return 0;
//            }
//            return 0;
//        });

        /**
         * 线程串行化
         * 1）、thenRun：不能获取到上一步的执行结果，无返回值
         *      .thenRunAsync(() -> {
         *             System.out.println("任务2启动了……");
         *         });
         * 2）、thenAcceptAsync：能接受上一步结果，但是无返回值
         *      .thenAcceptAsync(t->{
         *             System.out.println("任务2启动了……"+t);
         *         },executor);
         * 3)、thenApplyAsync：能接收上一步结果，有返回值
         */
//        CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 2;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).thenApplyAsync(t -> {
//            System.out.println("任务2启动了……" + t);
//            return "Hello " + t;
//        }, executor);
//
//        String s = future.get();

        /**
         * 两个都完成
         */
//        CompletableFuture<Object> future01 = CompletableFuture.supplyAsync(() -> {
//            System.out.println("任务1线程：" + Thread.currentThread().getId());
//            int i = 10 / 2;
//            System.out.println("任务1结束：");
//            return i;
//        }, executor);
//        CompletableFuture<Object> future02 = CompletableFuture.supplyAsync(() -> {
//            System.out.println("任务2线程：" + Thread.currentThread().getId());
//
//            try {
//                Thread.sleep(3000);
//                System.out.println("任务2结束：");
//            } catch (InterruptedException e) {
//                e.printStackTrace();
//            }
//            return "Hello";
//        }, executor);

//        future01.runAfterBothAsync(future02,()->{
//            System.out.println("任务3开始……");
//        },executor);

//        future01.thenAcceptBothAsync(future02,(t,u)->{
//            System.out.println("任务3开始……之前的结果："+t+","+u);
//        },executor);

//        CompletableFuture<String> future = future01.thenCombineAsync(future02, (t, u) -> {
//            return t + ":" + u + "->Haha";
//        }, executor);

        /**
         * 两个任务，只要有一个完成，我们就执行任务3
         * runAfterEitherAsync：不感知结果，自己也无返回值
         * acceptEitherAsync：感知结果，自己没有返回值
         * applyToEitherAsync：感知结果，有返回值
         */
//        future01.runAfterEitherAsync(future02,()->{
//            System.out.println("任务3开始……");
//        },executor);

//        future01.acceptEitherAsync(future02,(t)->{
//            System.out.println("任务3开始……"+t);
//        },executor);

//        CompletableFuture<String> future = future01.applyToEitherAsync(future02, (t) -> {
//            System.out.println("任务3开始……"+t);
//            return t + "哈哈";
//        }, executor);

        CompletableFuture<String> futureImg = CompletableFuture.supplyAsync(() -> {
            System.out.println("查看商品图片信息");
            return "hello.jpg";
        },executor);
        CompletableFuture<String> futureAttr = CompletableFuture.supplyAsync(() -> {
            System.out.println("查看商品的属性");
            return "黑色+256G";
        },executor);
        CompletableFuture<String> futureDesc = CompletableFuture.supplyAsync(() -> {

            try {
                Thread.sleep(3000);
                System.out.println("查看商品介绍");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "华为";
        },executor);

//        CompletableFuture<Void> allOf = CompletableFuture.allOf(futureImg, futureAttr, futureDesc);
//        allOf.get();//等待所有结果完成

        CompletableFuture<Object> anyOf = CompletableFuture.anyOf(futureImg, futureAttr, futureDesc);
        anyOf.get();
        System.out.println("main……end……");
    }


    public void thread(String[] args) throws ExecutionException, InterruptedException {

//        Thread01 thread=new Thread01();
//        thread.start();
//        System.out.println("main……end……");

//        Runnable01 runnable01=new Runnable01();
//        new Thread(runnable01).start();
//        System.out.println("main……end……");

//        FutureTask<Integer> futureTask = new FutureTask<>(new Callable01());
//        new Thread(futureTask).start();
//        //阻塞等待整个线程执行完，获取返回结果
//        Integer integer = futureTask.get();
//        System.out.println("main……end……"+integer);

        /**
         * corePoolSize:核心线程数[一直存在除非(allowCoreThreadTimeOut)]；线程池，创建好以后就准备就绪的线程数量，就等待来接收异步任务去执行
         * maximumPoolSize:最大线程数量；控制资源
         * keepAliveTime:存活时间。如果当前的线程数量大于core数量
         *      释放空闲线程（maximumPoolSize-corePoolSize）。只要线程空闲时间大于指定的keepAliveTime
         * unit:时间单位
         * BlockingQueue<Runnable> workQueue:阻塞队列。如果任务有很多，就会将目前多的任务放到队列里面
         *      只要有线程空闲。就会去队列里面取出新任务继续执行
         * threadFactory:线程的创建工厂
         * RejectedExecutionHandler handler:如果队列满了，按照我们制定的拒绝策略拒绝执行任务
         *
         *
         *
         * 工作顺序：
         * 1）、线程池创建。准备好core数量的核心线程，准备接受任务
         * 1.1、core满了，就将再进来的任务放入阻塞队列中，空闲的core就会自己去阻塞队列获取任务执行
         * 1.2、阻塞队列满了，就直接开新线程执行，最大只能开到max指定的数量
         * 1.3、max满了就用RejectedExecutionHandler拒绝任务
         * 1.4、max都执行完成，有很多空闲，在指定的keepAliveTime时间以后，释放max-core这些线程
         *
         *      new LinkedBlockingDeque<>()：默认是integer的最大值。内存不够
         *
         * 一个线程池core 7；max 20 , queue 50, 100并发进来怎么分配
         * 7个会立即得到执行，50个会进入队列，再开13个进行执行。剩下的30个就使用拒绝策略
         * 如果不想抛弃还要执行。CallerRunsPolicy；
         */
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(5,
                200,
                10,
                TimeUnit.SECONDS,
                new LinkedBlockingDeque<>(100000),
                Executors.defaultThreadFactory(),
                new ThreadPoolExecutor.AbortPolicy());

//        Executors.newCachedThreadPool();  core是0，所有都可回收
//        Executors.newFixedThreadPool();  固定大小，core=max，都不可回收
//        Executors.newScheduledThreadPool();  定时任务的线程池
//        Executors.newSingleThreadExecutor();   单线程的线程池，后台从队列里面获取到任务，挨个执行

    }
    //继承Thread
    public static class Thread01 extends Thread{
        @Override
        public void run() {
            System.out.println("当前线程："+Thread.currentThread().getId());
            int i=10/2;
            System.out.println("运行结果："+i);
        }
    }
    //实现Runnable接口
    public static class Runnable01 implements Runnable{

        @Override
        public void run() {
            System.out.println("当前线程："+Thread.currentThread().getId());
            int i=10/2;
            System.out.println("运行结果："+i);
        }
    }
    //实现Callable接口+FutureTask
    public static class Callable01 implements Callable<Integer>{

        @Override
        public Integer call() throws Exception {
            System.out.println("当前线程："+Thread.currentThread().getId());
            int i=10/2;
            System.out.println("运行结果："+i);
            return i;
        }
    }
}
